// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

using Agent.Sdk;
using Agent.Sdk.Knob;
using Microsoft.TeamFoundation.DistributedTask.WebApi;
using Microsoft.VisualStudio.Services.Agent.Util;
using System;
using System.Collections.Generic;
using System.Globalization;
using System.IO;
using System.Linq;

namespace Microsoft.VisualStudio.Services.Agent.Worker
{
    public sealed class TaskCommandExtension: BaseWorkerCommandExtension
    {
        public TaskCommandExtension()
        {
            CommandArea = "task";
            SupportedHostTypes = HostTypes.All;
            InstallWorkerCommand(new TaskIssueCommand());
            InstallWorkerCommand(new TaskProgressCommand());
            InstallWorkerCommand(new TaskDetailCommand());
            InstallWorkerCommand(new TaskCompleteCommand());
            InstallWorkerCommand(new TaskSetSecretCommand());
            InstallWorkerCommand(new TaskSetVariableCommand());
            InstallWorkerCommand(new TaskAddAttachmentCommand());
            InstallWorkerCommand(new TaskDebugCommand());
            InstallWorkerCommand(new TaskUploadSummaryCommand());
            InstallWorkerCommand(new TaskUploadFileCommand());
            InstallWorkerCommand(new TaskSetTaskVariableCommand());
            InstallWorkerCommand(new TaskSetEndpointCommand());
            InstallWorkerCommand(new TaskPrepandPathCommand());
        }
    }

    [CommandRestriction(AllowedInRestrictedMode=true)]
    public sealed class TaskDetailCommand: IWorkerCommand
    {
        public string Name => "logdetail";
        public List<string> Aliases => null;

        // Since we process all logging command in serialized order, everthing should be thread safe.
        private readonly Dictionary<Guid, TimelineRecord> _timelineRecordsTracker = new Dictionary<Guid, TimelineRecord>();
        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var eventProperties = command.Properties;
            var data = command.Data;

            TimelineRecord record = new TimelineRecord();

            String timelineRecord;
            if (!eventProperties.TryGetValue(TaskDetailEventProperties.TimelineRecordId, out timelineRecord) ||
                string.IsNullOrEmpty(timelineRecord) ||
                new Guid(timelineRecord).Equals(Guid.Empty))
            {
                throw new ArgumentNullException(StringUtil.Loc("MissingTimelineRecordId"));
            }
            else
            {
                record.Id = new Guid(timelineRecord);
            }

            string parentTimlineRecord;
            if (eventProperties.TryGetValue(TaskDetailEventProperties.ParentTimelineRecordId, out parentTimlineRecord))
            {
                record.ParentId = new Guid(parentTimlineRecord);
            }

            String name;
            if (eventProperties.TryGetValue(TaskDetailEventProperties.Name, out name))
            {
                record.Name = name;
            }

            String recordType;
            if (eventProperties.TryGetValue(TaskDetailEventProperties.Type, out recordType))
            {
                record.RecordType = recordType;
            }

            String order;
            if (eventProperties.TryGetValue(TaskDetailEventProperties.Order, out order))
            {
                int orderInt = 0;
                if (int.TryParse(order, out orderInt))
                {
                    record.Order = orderInt;
                }
            }

            String percentCompleteValue;
            if (eventProperties.TryGetValue(TaskDetailEventProperties.Progress, out percentCompleteValue))
            {
                Int32 progress;
                if (Int32.TryParse(percentCompleteValue, out progress))
                {
                    record.PercentComplete = (Int32)Math.Min(Math.Max(progress, 0), 100);
                }
            }

            if (!String.IsNullOrEmpty(data))
            {
                record.CurrentOperation = data;
            }

            string result;
            if (eventProperties.TryGetValue(TaskDetailEventProperties.Result, out result))
            {
                record.Result = EnumUtil.TryParse<TaskResult>(result) ?? TaskResult.Succeeded;
            }

            String startTime;
            if (eventProperties.TryGetValue(TaskDetailEventProperties.StartTime, out startTime))
            {
                record.StartTime = ParseDateTime(startTime, DateTime.UtcNow);
            }

            String finishtime;
            if (eventProperties.TryGetValue(TaskDetailEventProperties.FinishTime, out finishtime))
            {
                record.FinishTime = ParseDateTime(finishtime, DateTime.UtcNow);
            }

            String state;
            if (eventProperties.TryGetValue(TaskDetailEventProperties.State, out state))
            {
                record.State = ParseTimelineRecordState(state, TimelineRecordState.Pending);
            }


            TimelineRecord trackingRecord;
            // in front validation as much as possible.
            // timeline record is happened in back end queue, user will not receive result of the timeline record updates.
            // front validation will provide user better understanding when things went wrong.
            if (_timelineRecordsTracker.TryGetValue(record.Id, out trackingRecord))
            {
                // we already created this timeline record
                // make sure parentid does not changed.
                if (record.ParentId != null &&
                    record.ParentId != trackingRecord.ParentId)
                {
                    throw new InvalidOperationException(StringUtil.Loc("CannotChangeParentTimelineRecord"));
                }
                else if (record.ParentId == null)
                {
                    record.ParentId = trackingRecord.ParentId;
                }

                // populate default value for empty field.
                if (record.State == TimelineRecordState.Completed)
                {
                    if (record.PercentComplete == null)
                    {
                        record.PercentComplete = 100;
                    }

                    if (record.FinishTime == null)
                    {
                        record.FinishTime = DateTime.UtcNow;
                    }
                }
            }
            else
            {
                // we haven't created this timeline record
                // make sure we have name/type and parent record has created.
                if (string.IsNullOrEmpty(record.Name))
                {
                    throw new ArgumentNullException(StringUtil.Loc("NameRequiredForTimelineRecord"));
                }

                if (string.IsNullOrEmpty(record.RecordType))
                {
                    throw new ArgumentNullException(StringUtil.Loc("TypeRequiredForTimelineRecord"));
                }

                if (record.ParentId != null && record.ParentId != Guid.Empty)
                {
                    if (!_timelineRecordsTracker.ContainsKey(record.ParentId.Value))
                    {
                        throw new ArgumentNullException(StringUtil.Loc("ParentTimelineNotCreated"));
                    }
                }

                // populate default value for empty field.
                if (record.StartTime == null)
                {
                    record.StartTime = DateTime.UtcNow;
                }

                if (record.State == null)
                {
                    record.State = TimelineRecordState.InProgress;
                }
            }

            context.UpdateDetailTimelineRecord(record);

            _timelineRecordsTracker[record.Id] = record;
        }

        private DateTime ParseDateTime(String dateTimeText, DateTime defaultValue)
        {
            DateTime dateTime;
            if (!DateTime.TryParse(dateTimeText, CultureInfo.CurrentCulture, DateTimeStyles.AdjustToUniversal, out dateTime))
            {
                dateTime = defaultValue;
            }

            return dateTime;
        }

        private TimelineRecordState ParseTimelineRecordState(String timelineRecordStateText, TimelineRecordState defaultValue)
        {
            TimelineRecordState state;
            if (!Enum.TryParse<TimelineRecordState>(timelineRecordStateText, out state))
            {
                state = defaultValue;
            }

            return state;
        }
    }

    public sealed class TaskUploadSummaryCommand: IWorkerCommand
    {
        public string Name => "uploadsummary";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var data = command.Data;
            if (!string.IsNullOrEmpty(data))
            {
                var uploadSummaryProperties = new Dictionary<string, string>();
                uploadSummaryProperties.Add(TaskAddAttachmentEventProperties.Type, CoreAttachmentType.Summary);
                var fileName = Path.GetFileName(data);
                uploadSummaryProperties.Add(TaskAddAttachmentEventProperties.Name, fileName);

                TaskAddAttachmentCommand.AddAttachment(context, uploadSummaryProperties, data);
            }
            else
            {
                throw new InvalidOperationException(StringUtil.Loc("CannotUploadSummary"));
            }
        }
    }

    public sealed class TaskUploadFileCommand: IWorkerCommand
    {
        public string Name => "uploadfile";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var data = command.Data;

            if (!string.IsNullOrEmpty(data))
            {
                var uploadFileProperties = new Dictionary<string, string>();
                uploadFileProperties.Add(TaskAddAttachmentEventProperties.Type, CoreAttachmentType.FileAttachment);
                var fileName = Path.GetFileName(data);
                uploadFileProperties.Add(TaskAddAttachmentEventProperties.Name, fileName);

                TaskAddAttachmentCommand.AddAttachment(context, uploadFileProperties, data);
            }
            else
            {
                throw new InvalidOperationException(StringUtil.Loc("CannotUploadFile"));
            }
        }
    }

    public sealed class TaskAddAttachmentCommand: IWorkerCommand
    {
        public string Name => "addattachment";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            AddAttachment(context, command.Properties, command.Data);
        }

        public static void AddAttachment(IExecutionContext context, Dictionary<string, string> eventProperties, string data)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(eventProperties, nameof(eventProperties));

            String type;
            if (!eventProperties.TryGetValue(TaskAddAttachmentEventProperties.Type, out type) || String.IsNullOrEmpty(type))
            {
                throw new ArgumentNullException(StringUtil.Loc("MissingAttachmentType"));
            }

            String name;
            if (!eventProperties.TryGetValue(TaskAddAttachmentEventProperties.Name, out name) || String.IsNullOrEmpty(name))
            {
                throw new ArgumentNullException(StringUtil.Loc("MissingAttachmentName"));
            }

            char[] s_invalidFileChars = Path.GetInvalidFileNameChars();
            if (type.IndexOfAny(s_invalidFileChars) != -1)
            {
                throw new ArgumentException($"Type contains invalid characters. ({String.Join(",", s_invalidFileChars)})");
            }

            if (name.IndexOfAny(s_invalidFileChars) != -1)
            {
                throw new ArgumentException($"Name contains invalid characters. ({String.Join(", ", s_invalidFileChars)})");
            }

            // Translate file path back from container path
            string filePath = context.TranslateToHostPath(data);

            if (!String.IsNullOrEmpty(filePath) && File.Exists(filePath))
            {
                // Upload attachment
                context.QueueAttachFile(type, name, filePath);
            }
            else
            {
                throw new ArgumentNullException(StringUtil.Loc("MissingAttachmentFile"));
            }
        }
    }

    [CommandRestriction(AllowedInRestrictedMode=true)]
    public sealed class TaskIssueCommand: IWorkerCommand
    {
        public string Name => "logissue";
        public List<string> Aliases => new List<string>(){"issue"};

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var eventProperties = command.Properties;
            var data = command.Data;

            Issue taskIssue = null;

            String issueType;
            if (eventProperties.TryGetValue(TaskIssueEventProperties.Type, out issueType))
            {
                taskIssue = CreateIssue(context, issueType, data, eventProperties);
            }

            if (taskIssue == null)
            {
                context.Warning("Can't create TaskIssue from logging event.");
                return;
            }

            context.AddIssue(taskIssue);
        }

        private Issue CreateIssue(IExecutionContext context, string issueType, String message, Dictionary<String, String> properties)
        {
            Issue issue = new Issue()
            {
                Category = "General",
            };

            if (issueType.Equals("error", StringComparison.OrdinalIgnoreCase))
            {
                issue.Type = IssueType.Error;
            }
            else if (issueType.Equals("warning", StringComparison.OrdinalIgnoreCase))
            {
                issue.Type = IssueType.Warning;
            }
            else
            {
                throw new ArgumentException($"issue type {issueType} is not an expected issue type.");
            }

            String sourcePath;
            if (properties.TryGetValue(ProjectIssueProperties.SourcePath, out sourcePath))
            {
                issue.Category = "Code";

                var extensionManager = context.GetHostContext().GetService<IExtensionManager>();
                var hostType = context.Variables.System_HostType;
                IJobExtension extension =
                    (extensionManager.GetExtensions<IJobExtension>() ?? new List<IJobExtension>())
                    .Where(x => x.HostType.HasFlag(hostType))
                    .FirstOrDefault();

                if (extension != null)
                {
                    // Translate file path back from container path
                    sourcePath = context.TranslateToHostPath(sourcePath);
                    properties[ProjectIssueProperties.SourcePath] = sourcePath;

                    // Get the values that represent the server path given a local path
                    string repoName;
                    string relativeSourcePath;
                    extension.ConvertLocalPath(context, sourcePath, out repoName, out relativeSourcePath);

                    // add repo info
                    if (!string.IsNullOrEmpty(repoName))
                    {
                        properties["repo"] = repoName;
                    }

                    if (!string.IsNullOrEmpty(relativeSourcePath))
                    {
                        // replace sourcePath with the new relative path
                        properties[ProjectIssueProperties.SourcePath] = relativeSourcePath;
                    }

                    String sourcePathValue;
                    String lineNumberValue;
                    String columnNumberValue;
                    String messageTypeValue;
                    String codeValue;
                    properties.TryGetValue(TaskIssueEventProperties.Type, out messageTypeValue);
                    properties.TryGetValue(ProjectIssueProperties.SourcePath, out sourcePathValue);
                    properties.TryGetValue(ProjectIssueProperties.LineNumber, out lineNumberValue);
                    properties.TryGetValue(ProjectIssueProperties.ColumNumber, out columnNumberValue);
                    properties.TryGetValue(ProjectIssueProperties.Code, out codeValue);

                    //ex. Program.cs(13, 18): error CS1002: ; expected
                    message = String.Format(CultureInfo.InvariantCulture, "{0}({1},{2}): {3} {4}: {5}",
                        sourcePathValue,
                        lineNumberValue,
                        columnNumberValue,
                        messageTypeValue,
                        codeValue,
                        message);
                }
            }

            if (properties != null)
            {
                foreach (var property in properties)
                {
                    issue.Data[property.Key] = property.Value;
                }
            }

            issue.Message = message;

            return issue;
        }
    }

    [CommandRestriction(AllowedInRestrictedMode=true)]
    public sealed class TaskCompleteCommand: IWorkerCommand
    {
        public string Name => "complete";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var eventProperties = command.Properties;
            var data = command.Data;

            string resultText;
            TaskResult result;
            if (!eventProperties.TryGetValue(TaskCompleteEventProperties.Result, out resultText) ||
                String.IsNullOrEmpty(resultText) ||
                !Enum.TryParse<TaskResult>(resultText, out result))
            {
                throw new ArgumentException(StringUtil.Loc("InvalidCommandResult"));
            }

            context.Result = TaskResultUtil.MergeTaskResults(context.Result, result);
            context.Progress(100, data);

            if (eventProperties.TryGetValue(TaskCompleteEventProperties.Done, out string doneText) &&
                !String.IsNullOrEmpty(doneText) &&
                StringUtil.ConvertToBoolean(doneText))
            {
                context.ForceTaskComplete();
            }
        }
    }

    [CommandRestriction(AllowedInRestrictedMode=true)]
    public sealed class TaskProgressCommand: IWorkerCommand
    {
        public string Name => "setprogress";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var eventProperties = command.Properties;
            var data = command.Data;

            Int32 percentComplete = 0;
            String processValue;
            if (eventProperties.TryGetValue("value", out processValue))
            {
                Int32 progress;
                if (Int32.TryParse(processValue, out progress))
                {
                    percentComplete = (Int32)Math.Min(Math.Max(progress, 0), 100);
                }
            }

            context.Progress(percentComplete, data);
        }
    }

    [CommandRestriction(AllowedInRestrictedMode=true)]
    public sealed class TaskSetSecretCommand: IWorkerCommand
    {
        public string Name => "setsecret";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var data = command.Data;
            if (!string.IsNullOrEmpty(data))
            {
                context.GetHostContext().SecretMasker.AddValue(data);
            }
        }
    }

    [CommandRestriction(AllowedInRestrictedMode=true)]
    public sealed class TaskSetVariableCommand: IWorkerCommand
    {
        public string Name => "setvariable";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var eventProperties = command.Properties;
            var data = command.Data;

            String name;
            if (!eventProperties.TryGetValue(TaskSetVariableEventProperties.Variable, out name) || String.IsNullOrEmpty(name))
            {
                throw new ArgumentNullException(StringUtil.Loc("MissingVariableName"));
            }

            String isSecretValue;
            Boolean isSecret = false;
            if (eventProperties.TryGetValue(TaskSetVariableEventProperties.IsSecret, out isSecretValue))
            {
                Boolean.TryParse(isSecretValue, out isSecret);
            }

            String isOutputValue;
            Boolean isOutput = false;
            if (eventProperties.TryGetValue(TaskSetVariableEventProperties.IsOutput, out isOutputValue))
            {
                Boolean.TryParse(isOutputValue, out isOutput);
            }

            String isReadOnlyValue;
            Boolean isReadOnly = false;
            if (eventProperties.TryGetValue(TaskSetVariableEventProperties.IsReadOnly, out isReadOnlyValue))
            {
                Boolean.TryParse(isReadOnlyValue, out isReadOnly);
            }

            if (context.Variables.IsReadOnly(name))
            {
                // Check FF. If it is on then throw, otherwise warn
                // TODO - remove this and just always throw once the feature has been fully rolled out.
                if (context.Variables.Read_Only_Variables)
                {
                    throw new InvalidOperationException(StringUtil.Loc("ReadOnlyVariable", name));
                }
                else
                {
                    context.Warning(StringUtil.Loc("ReadOnlyVariableWarning", name));
                }
            }

            if (isSecret)
            {

                if (!string.IsNullOrEmpty(data) &&
                    data.Contains(Environment.NewLine) &&
                    !AgentKnobs.AllowUnsafeMultilineSecret.GetValue(context).AsBoolean())
                {
                    throw new InvalidOperationException(StringUtil.Loc("MultilineSecret"));
                }

                var unescapePercents = AgentKnobs.DecodePercents.GetValue(context).AsBoolean();
                var commandEscapeData = CommandStringConvertor.Escape(command.Data, unescapePercents);
                context.GetHostContext().SecretMasker.AddValue(commandEscapeData);
            }

            var checker = context.GetHostContext().GetService<ITaskRestrictionsChecker>();
            if (checker.CheckSettableVariable(context, name))
            {
                context.SetVariable(name, data, isSecret: isSecret, isOutput: isOutput, isReadOnly: isReadOnly);
            }
        }
    }

    [CommandRestriction(AllowedInRestrictedMode=true)]
    public sealed class TaskDebugCommand: IWorkerCommand
    {
        public string Name => "debug";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var data = command.Data;
            context.Debug(data);
        }
    }

    [CommandRestriction(AllowedInRestrictedMode=true)]
    public sealed class TaskSetTaskVariableCommand: IWorkerCommand
    {
        public string Name => "settaskvariable";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var eventProperties = command.Properties;
            var data = command.Data;

            String name;
            if (!eventProperties.TryGetValue(TaskSetTaskVariableEventProperties.Variable, out name) || String.IsNullOrEmpty(name))
            {
                throw new ArgumentNullException(StringUtil.Loc("MissingTaskVariableName"));
            }

            String isSecretValue;
            Boolean isSecret = false;
            if (eventProperties.TryGetValue(TaskSetTaskVariableEventProperties.IsSecret, out isSecretValue))
            {
                Boolean.TryParse(isSecretValue, out isSecret);
            }

            String isReadOnlyValue;
            Boolean isReadOnly = false;
            if (eventProperties.TryGetValue(TaskSetVariableEventProperties.IsReadOnly, out isReadOnlyValue))
            {
                Boolean.TryParse(isReadOnlyValue, out isReadOnly);
            }

            if (context.TaskVariables.IsReadOnly(name))
            {
                // Check FF. If it is on then throw, otherwise warn
                // TODO - remove this and just always throw once the feature has been fully rolled out.
                if (context.Variables.Read_Only_Variables)
                {
                    throw new InvalidOperationException(StringUtil.Loc("ReadOnlyTaskVariable", name));
                }
                else
                {
                    context.Warning(StringUtil.Loc("ReadOnlyTaskVariableWarning", name));
                }
            }

            if (isSecret)
            {
                if (!string.IsNullOrEmpty(data) &&
                    data.Contains(Environment.NewLine) &&
                    !AgentKnobs.AllowUnsafeMultilineSecret.GetValue(context).AsBoolean())
                {
                    throw new InvalidOperationException(StringUtil.Loc("MultilineSecret"));
                }
            }

            context.TaskVariables.Set(name, data, secret: isSecret, readOnly: isReadOnly);
        }
    }

    public sealed class TaskSetEndpointCommand: IWorkerCommand
    {
        public string Name => "setendpoint";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var eventProperties = command.Properties;
            var data = command.Data;

            if (string.IsNullOrEmpty(data))
            {
                throw new ArgumentNullException(StringUtil.Loc("EnterValidValueFor0", "setendpoint"));
            }

            String field;
            if (!eventProperties.TryGetValue(TaskSetEndpointEventProperties.Field, out field) || String.IsNullOrEmpty(field))
            {
                throw new ArgumentNullException(StringUtil.Loc("MissingEndpointField"));
            }

            // Mask auth parameter data upfront to avoid accidental secret exposure by invalid endpoint/key/data
            if (String.Equals(field, "authParameter", StringComparison.OrdinalIgnoreCase))
            {
                context.GetHostContext().SecretMasker.AddValue(data);
            }

            String endpointIdInput;
            if (!eventProperties.TryGetValue(TaskSetEndpointEventProperties.EndpointId, out endpointIdInput) || String.IsNullOrEmpty(endpointIdInput))
            {
                throw new ArgumentNullException(StringUtil.Loc("MissingEndpointId"));
            }

            Guid endpointId;
            if (!Guid.TryParse(endpointIdInput, out endpointId))
            {
                throw new ArgumentNullException(StringUtil.Loc("InvalidEndpointId"));
            }

            var endpoint = context.Endpoints.Find(a => a.Id == endpointId);
            if (endpoint == null)
            {
                throw new ArgumentNullException(StringUtil.Loc("InvalidEndpointId"));
            }

            if (String.Equals(field, "url", StringComparison.OrdinalIgnoreCase))
            {
                Uri uri;
                if (!Uri.TryCreate(data, UriKind.Absolute, out uri))
                {
                    throw new ArgumentNullException(StringUtil.Loc("InvalidEndpointUrl"));
                }

                endpoint.Url = uri;
                return;
            }

            String key;
            if (!eventProperties.TryGetValue(TaskSetEndpointEventProperties.Key, out key) || String.IsNullOrEmpty(key))
            {
                throw new ArgumentNullException(StringUtil.Loc("MissingEndpointKey"));
            }

            if (String.Equals(field, "dataParameter", StringComparison.OrdinalIgnoreCase))
            {
                endpoint.Data[key] = data;
            }
            else if (String.Equals(field, "authParameter", StringComparison.OrdinalIgnoreCase))
            {
                endpoint.Authorization.Parameters[key] = data;
            }
            else
            {
                throw new ArgumentException(StringUtil.Loc("InvalidEndpointField"));
            }
        }
    }

    [CommandRestriction(AllowedInRestrictedMode=true)]
    public sealed class TaskPrepandPathCommand: IWorkerCommand
    {
        public string Name => "prependpath";
        public List<string> Aliases => null;

        public void Execute(IExecutionContext context, Command command)
        {
            ArgUtil.NotNull(context, nameof(context));
            ArgUtil.NotNull(command, nameof(command));

            var checker = context.GetHostContext().GetService<ITaskRestrictionsChecker>();
            if (!checker.CheckSettableVariable(context, Constants.PathVariable))
            {
                return;
            }

            var data = command.Data;

            ArgUtil.NotNullOrEmpty(data, this.Name);
            context.PrependPath.RemoveAll(x => string.Equals(x, data, StringComparison.CurrentCulture));
            context.PrependPath.Add(data);
        }
    }


    internal static class TaskSetVariableEventProperties
    {
        public static readonly String Variable = "variable";
        public static readonly String IsSecret = "issecret";
        public static readonly String IsOutput = "isoutput";
        public static readonly String IsReadOnly = "isreadonly";
    }

    internal static class TaskCompleteEventProperties
    {
        public static readonly String Result = "result";
        public static readonly String Done = "done";
    }

    internal static class TaskIssueEventProperties
    {
        public static readonly String Type = "type";
    }

    internal static class ProjectIssueProperties
    {
        public static readonly String Code = "code";
        public static readonly String ColumNumber = "columnnumber";
        public static readonly String SourcePath = "sourcepath";
        public static readonly String LineNumber = "linenumber";
    }

    internal static class TaskAddAttachmentEventProperties
    {
        public static readonly String Type = "type";
        public static readonly String Name = "name";
    }

    internal static class TaskDetailEventProperties
    {
        public static readonly String TimelineRecordId = "id";
        public static readonly String ParentTimelineRecordId = "parentid";
        public static readonly String Type = "type";
        public static readonly String Name = "name";
        public static readonly String StartTime = "starttime";
        public static readonly String FinishTime = "finishtime";
        public static readonly String Progress = "progress";
        public static readonly String State = "state";
        public static readonly String Result = "result";
        public static readonly String Order = "order";
    }

    internal static class TaskSetTaskVariableEventProperties
    {
        public static readonly String Variable = "variable";
        public static readonly String IsSecret = "issecret";
    }

    internal static class TaskSetEndpointEventProperties
    {
        public static readonly String EndpointId = "id";
        public static readonly String Field = "field";
        public static readonly String Key = "key";
    }
}
